| The Media Kit Table of Contents | The Media Kit Index | 
Derived from: virtual BMediaNode
Declared in: be/media/Controllable.h
Library: libmedia.so
Allocation: Constructor only
A node whose behavior can be controlled by the user should be derived from BControllable, as well as from whatever other interface classes the node might be derived from. Deriving from BControllable lets the node publish information about the parameters that can be adjusted and how they relate to each other.
A client application can use the published information to build a user interface for the node or pass the information through to a system routine that will build the interface.
A node can also have the ability to start its own control panel under outside control, which can take advantage of special knowledge of the node.
protected:
| 
 | 
The BControllable constructor. You should override this in your derived class to create a BParameterWeb object, configure it to describe the available parameters, and call BControllable::SetParameterWeb() with that object before returning.
protected:
| 
 | 
When the configuration of a specific parameter changes, you should call this function with the ID of the changed parameter so that clients know that they need to check with the node to determine the parameter's new configuration.
| 
 | 
RETURN CODES
B_OK. No errors.
protected:
| 
 | 
Call this function when a parameter value change takes effect and you want people that are interested in knowing about the change to stay in sync with you. Unlike BroadcastChangedParameter(), this function actually passes along the new value of the parameter.
The when argument indicates the performance time at which the change took effect. The id indicates the parameter ID of the parameter whose value changed. newValue is a pointer to the parameter's data, and valueSize defines the size of that data.
RETURN CODES
B_OK. No error.
protected:
| 
 | 
You should implement GetParameterValue() to store the value of the parameter with the specified parameterID in the memory pointed to by value. The size_t value pointed to by ioSize specifies the size of the value buffer; prior to returning, your implementation of GetParameterValue() should change ioSize to the actual size of the returned data.
Also, you should set lastChangeTime to the time at which the control's value most recently changed.
GetParameterValue() should return B_OK when done, or an appropriate error code if something goes wrong.
Likewise, you should implement SetParameterValue() to change the value of the parameter; the changeTime argument is the performance time at which the change should occur; in other words, you may need to queue the request so it can be handled at the requested time. value points to the value to which the parameter should be set, and size is the number of bytes of data in the value.
| 
 | 
| 
 | 
When your node's service loop receives a message, in addition to passing it to BMediaNode and other superclasses of your node, you should also pass it to BControllable::HandleMessage(). You should start at the most-derived class' implementation of HandleMessage() and work your way upward until B_OK is returned.
If it's a message intended for the BControllable interface, it'll be dispatched and B_OK will be returned; otherwise, HandleMessage() will return an error so you know to try something else.
   /* Message received */
   
   if ((BControllable::HandleMessage(message, data, size) != B_OK) &&
            (BMediaNode::HandleMessage(message, data, size) != B_OK)) {
      BMediaNode::HandleBadMessage(message, data, size);
   }
In this example, the BControllable implementation of HandleMessage() gets the first crack at handling the request. If it doesn't know what to do with the message, it's forwarded to BMediaNode's implementation. If the message still isn't handled, it's then sent to BMediaNode::HandleBadMessage() to be dealt with.
RETURN CODES
B_OK. The message was dispatched.
See also: BMediaNode::HandleMessage(), "About Multiple Virtual Inheritance" on page21
| 
 | 
LockParameterWeb() locks the web to prevent access to it while you're working on it, and UnlockParameterWeb() releases it when you're done. You should surround your accesses to the web with these calls:
   LockParameterWeb();
   Web()->MakeGroup("EqualizerControls");
   ...
   UnlockParameterWeb();
protected:
| 
 | 
The MakeParameterData() utility function takes a list of parameter IDs from parameterList and calls GetParameterValue() for each of them, storing the values in the specified buffer until the size specified in ioSize is filled, or all the parameters are read. The number of bytes of the buffer used will be returned in ioSize.
If your BControllable is also a BBufferConsumer that accepts B_MEDIA_PARAMETERS type data on some input, call ApplyParameterData() with value set to the result of BBuffer::Data() and size set to BBuffer::Size(). This function will then parse the parameter change requests in the buffer and dispatch them to your SetParameterValue() function to fulfill the requests.
This lets your node support easy automation of parameter information. Even more benefit can be obtained by also deriving from BBufferProducer, and providing an output for the B_MEDIA_PARAMETERS data format, so that changes can be recorded as they occur. This provides a mechanism for automating the parameters by recording a user's changes to them, then playing back the changes later.
RETURN CODES
B_OK. No errors.
protected:
| 
 | 
Your constructor should create a BParameterWeb object and call SetParameterWeb() with it as an argument. This will describe to the outside world what parameters are available and how they relate to each other; in other words, this describes your internal signal path, and how it can be manipulated.
If the web argument isn't NULL, and is different from the previously-established web for the BControllable node, a B_MEDIA_WEB_CHANGED message is sent to everyone watching for media notifications. See StartWatching() on for more information.
SetParameterWeb() will return B_OK if the web was set without errors; otherwise an error code will be returned.
The Web() function returns the BParameterWeb assigned to the BControllable.
protected:
| 
 | 
This hook function is called whenever a client application wants the node to present its own control panel user interface (so that the user can configure the node).
On return, outMessenger is a BMessenger that you can use to communicate with teh control panel.
Because the add-on lives in the Media Server, and a problem in the user interface could bring down the entire system, it's recommended that the control panel run as its own team. This can be done easily by writing your node as both a Media Server add-on (by exporting make_media_addon()) and an application (by implementing main() and including start_dyn.o among the link libraries). Be sure you have the multi-launch application flags set on your add-on, or this won't work right.
Then your StartControlPanel() implementation can simply launch the add-on as an application, and if the user double-clicks the add-on, they'll be presented with the control panel. In addition, the user benefits by only having to install a single file for your add-on to work properly.
The first argv argument to your main() function will be a string of the format "node=%d" with the node ID in question as "%d".
| 
 | 
RETURN CODES
B_OK. No errors occurred.
| The Media Kit Table of Contents | The Media Kit Index | 
Copyright © 2000 Be, Inc. All rights reserved..